<template>
    <view
        class="u-switch"
        :class="[modelValue == true ? 'u-switch--on' : '', disabled ? 'u-switch--disabled' : '', customClass]"
        @tap="onClick"
        :style="$u.toStyle(switchStyle, customStyle)"
    >
        <view
            class="u-switch__node node-class"
            :style="{
                width: $u.addUnit(size ?? 50),
                height: $u.addUnit(size ?? 50)
            }"
        >
            <u-loading :show="loading" class="u-switch__loading" :size="Number(size) * 0.6" :color="loadingColor" />
        </view>
    </view>
</template>

<script lang="ts">
export default {
    name: 'u-switch',
    options: {
        addGlobalClass: true,
        // #ifndef MP-TOUTIAO
        virtualHost: true,
        // #endif
        styleIsolation: 'shared'
    }
};
</script>

<script setup lang="ts">
import { computed, nextTick } from 'vue';
import { $u } from '../..';
import { SwitchProps } from './types';

/**
 * switch 开关选择器
 * @description 选择开关一般用于只有两个选择，且只能选其一的场景。
 * @tutorial https://uviewpro.cn/zh/components/switch.html
 * @property {Boolean} loading 是否处于加载中（默认false）
 * @property {Boolean} disabled 是否禁用（默认false）
 * @property {String|Number} size 开关尺寸，单位rpx（默认50）
 * @property {String} active-color 打开时的背景色（默认主题色primary）
 * @property {String} inactive-color 关闭时的背景色（默认#ffffff）
 * @property {Boolean|Number|String} active-value 打开选择器时通过change事件发出的值（默认true）
 * @property {Boolean|Number|String} inactive-value 关闭选择器时通过change事件发出的值（默认false）
 * @event {Function} change 在switch打开或关闭时触发
 * @example <u-switch v-model="checked" active-color="red" inactive-color="#eee"></u-switch>
 */
const props = defineProps(SwitchProps);

const emit = defineEmits(['update:modelValue', 'change']);

/**
 * 计算开关样式
 */
const switchStyle = computed(() => {
    let style: Record<string, string> = {};
    style.fontSize = props.size + 'rpx';
    style.backgroundColor = props.modelValue ? props.activeColor : props.inactiveColor;
    return style;
});
/**
 * 计算加载动画颜色
 */
const loadingColor = computed(() => {
    return props.modelValue ? props.activeColor : null;
});

/**
 * 点击开关
 */
function onClick() {
    if (!props.disabled && !props.loading) {
        // 使手机产生短促震动，微信小程序有效，APP(HX 2.6.8)和H5无效
        if (props.vibrateShort) uni.vibrateShort();
        emit('update:modelValue', !props.modelValue);
        // 放到下一个生命周期，因为双向绑定的value修改父组件状态需要时间，且是异步的
        nextTick(() => {
            emit('change', props.modelValue ? props.activeValue : props.inactiveValue);
        });
    }
}
</script>

<style lang="scss" scoped>
@import '../../libs/css/style.components.scss';

.u-switch {
    position: relative;
    /* #ifndef APP-NVUE */
    display: inline-block;
    /* #endif */
    box-sizing: initial;
    width: 2em;
    height: 1em;
    background-color: #fff;
    border: 1px solid rgba(0, 0, 0, 0.1);
    border-radius: 1em;
    transition: background-color 0.3s;
    font-size: 50rpx;
}

.u-switch__node {
    @include vue-flex;
    align-items: center;
    justify-content: center;
    position: absolute;
    top: 0;
    left: 0;
    border-radius: 100%;
    z-index: 1;
    background-color: #fff;
    background-color: #fff;
    box-shadow:
        0 3px 1px 0 rgba(0, 0, 0, 0.05),
        0 2px 2px 0 rgba(0, 0, 0, 0.1),
        0 3px 3px 0 rgba(0, 0, 0, 0.05);
    box-shadow:
        0 3px 1px 0 rgba(0, 0, 0, 0.05),
        0 2px 2px 0 rgba(0, 0, 0, 0.1),
        0 3px 3px 0 rgba(0, 0, 0, 0.05);
    transition: transform 0.3s cubic-bezier(0.3, 1.05, 0.4, 1.05);
    transition:
        transform 0.3s cubic-bezier(0.3, 1.05, 0.4, 1.05),
        -webkit-transform 0.3s cubic-bezier(0.3, 1.05, 0.4, 1.05);
    transition: transform cubic-bezier(0.3, 1.05, 0.4, 1.05);
    transition: transform 0.3s cubic-bezier(0.3, 1.05, 0.4, 1.05);
}

.u-switch__loading {
    @include vue-flex;
    align-items: center;
    justify-content: center;
}

.u-switch--on {
    background-color: $u-type-primary;
}

.u-switch--on .u-switch__node {
    transform: translateX(100%);
}

.u-switch--disabled {
    opacity: 0.4;
}
</style>
